home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / gsfcmap.c < prev    next >
C/C++ Source or Header  |  1997-03-17  |  5KB  |  154 lines

  1. /* Copyright (C) 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gsfcmap.c */
  20. /* CMap character decoding */
  21. #include "gx.h"
  22. #include "gserrors.h"
  23. #include "gsstruct.h"
  24. #include "gxfcmap.h"
  25.  
  26. /* CMap structure descriptors */
  27. public_st_cmap();
  28. public_st_code_map();
  29. public_st_code_map_element();
  30. #define pcmap ((gx_code_map *)vptr)
  31. /* Because code maps can be elements of arrays, */
  32. /* their enum_ptrs procedure must never return 0 prematurely. */
  33. private ENUM_PTRS_BEGIN(code_map_enum_ptrs) return 0;
  34.     ENUM_PTR(0, gx_code_map, cmap);
  35.     case 1:
  36.       switch ( pcmap->type )
  37.         {
  38.         case cmap_glyph:
  39.           (*pcmap->cmap->mark_glyph)(pcmap->data.glyph,
  40.                      pcmap->cmap->mark_glyph_data);
  41.         default:
  42.           ENUM_RETURN(0);
  43.         case cmap_subtree:
  44.           ENUM_RETURN_PTR(gx_code_map, data.subtree);
  45.         }
  46. ENUM_PTRS_END
  47. private RELOC_PTRS_BEGIN(code_map_reloc_ptrs) ;
  48.     switch ( pcmap->type )
  49.       {
  50.       case cmap_subtree:
  51.         RELOC_PTR(gx_code_map, data.subtree);
  52.         break;
  53.       default:
  54.         ;
  55.       }
  56.     RELOC_PTR(gx_code_map, cmap);
  57. RELOC_PTRS_END
  58. #undef pcmap
  59. /* CIDSystemInfo structure */
  60. gs_public_st_composite(st_cid_system_info, gs_cid_system_info,
  61.   "gs_cid_system_info", cid_si_enum_ptrs, cid_si_reloc_ptrs);
  62. #define pcidsi ((gs_cid_system_info *)vptr)
  63. private ENUM_PTRS_BEGIN(cid_si_enum_ptrs) return 0;
  64.     ENUM_CONST_STRING_PTR(0, gs_cid_system_info, Registry);
  65.     ENUM_CONST_STRING_PTR(1, gs_cid_system_info, Ordering);
  66. ENUM_PTRS_END
  67. private RELOC_PTRS_BEGIN(cid_si_reloc_ptrs) ;
  68.     RELOC_CONST_STRING_PTR(gs_cid_system_info, Registry);
  69.     RELOC_CONST_STRING_PTR(gs_cid_system_info, Ordering);
  70. RELOC_PTRS_END
  71. #undef pcidsi
  72.  
  73. /* ---------------- Procedures ---------------- */
  74.  
  75. /*
  76.  * Decode a character from a string using a code map, updating the index.
  77.  * Return 0 for a CID or name, N > 0 for a character code where N is the
  78.  * number of bytes in the code, or an error.  For undefined characters,
  79.  * we set *pglyph = gs_no_glyph and return 0.
  80.  */
  81. private int
  82. code_map_decode_next(const gx_code_map *pcmap, const gs_const_string *str,
  83.   uint *pindex, uint *pfidx, gs_glyph *pglyph)
  84. {    const gx_code_map *map = pcmap;
  85.     uint chr = 0;
  86.  
  87.     for ( ; ; )
  88.       { int result;
  89.  
  90.         switch ( (gx_code_map_type)map->type )
  91.           {
  92.           case cmap_char_code:
  93.         *pglyph = (gs_glyph)map->data.ccode;
  94.         result = map->num_bytes1 + 1;
  95. leaf:        if ( chr > map->last )
  96.           goto undef;
  97.         if ( map->add_offset )
  98.           *pglyph += chr - map->first;
  99.         *pfidx = map->byte_data.font_index;
  100.         return result;
  101.           case cmap_glyph:
  102.         *pglyph = map->data.glyph;
  103.         result = 0;
  104.         goto leaf;
  105.           case cmap_subtree:
  106.         if ( *pindex >= str->size )
  107.           return_error(gs_error_rangecheck);
  108.         chr = str->data[(*pindex)++];
  109.         if ( chr >= map->data.subtree[0].first )
  110.           { /* Invariant: map[lo].first <= chr < map[hi].first. */
  111.             uint lo = 0, hi = map->byte_data.count1 + 1;
  112.  
  113.             map = map->data.subtree;
  114.             while ( lo + 1 < hi )
  115.               { uint mid = (lo + hi) >> 1;
  116.                 if ( chr >= map[mid].first )
  117.               lo = mid;
  118.             else
  119.               hi = mid;
  120.               }
  121.             map = &map[lo];
  122.             continue;
  123.           }
  124. undef:        *pglyph = gs_no_glyph;
  125.         return 0;
  126.           default:            /* (can't happen) */
  127.         return_error(gs_error_invalidfont);
  128.           }
  129.       }
  130. }
  131.  
  132. /*
  133.  * Decode a character from a string using a CMap.
  134.  * Return like code_map_decode_next.
  135.  */
  136. int
  137. gs_cmap_decode_next(const gs_cmap *pcmap, const gs_const_string *str,
  138.   uint *pindex, uint *pfidx, gs_glyph *pglyph)
  139. {    uint save_index = *pindex;
  140.     int code =
  141.       code_map_decode_next(&pcmap->def, str, pindex, pfidx, pglyph);
  142.  
  143.     if ( code != 0 || *pglyph != gs_no_glyph )
  144.       return code;
  145.     /* This is an undefined character.  Use the notdef map. */
  146.     { uint next_index = *pindex;
  147.       *pindex = save_index;
  148.       code =
  149.         code_map_decode_next(&pcmap->notdef, str, pindex, pfidx, pglyph);
  150.       *pindex = next_index;
  151.     }
  152.     return code;
  153. }
  154.